home *** CD-ROM | disk | FTP | other *** search
/ Aminet 2 / Aminet AMIGA CDROM (1994)(Walnut Creek)[Feb 1994][W.O. 44790-1].iso / Aminet / util / gnu / emacs_src.lha / emacs-18.58 / src / window.c < prev    next >
C/C++ Source or Header  |  1992-04-26  |  57KB  |  2,060 lines

  1. /* Window creation, deletion and examination for GNU Emacs.
  2.    Does not include redisplay.
  3.    Copyright (C) 1985, 1986, 1987, 1990 Free Software Foundation, Inc.
  4.  
  5. This file is part of GNU Emacs.
  6.  
  7. GNU Emacs is free software; you can redistribute it and/or modify
  8. it under the terms of the GNU General Public License as published by
  9. the Free Software Foundation; either version 1, or (at your option)
  10. any later version.
  11.  
  12. GNU Emacs is distributed in the hope that it will be useful,
  13. but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  15. GNU General Public License for more details.
  16.  
  17. You should have received a copy of the GNU General Public License
  18. along with GNU Emacs; see the file COPYING.  If not, write to
  19. the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
  20.  
  21.  
  22. #include "config.h"
  23. #include "lisp.h"
  24. #include "buffer.h"
  25. #include "window.h"
  26. #include "commands.h"
  27. #include "indent.h"
  28. #include "termchar.h"
  29.  
  30. Lisp_Object Qwindowp;
  31.  
  32. Lisp_Object Fnext_window (), Fdelete_window (), Fselect_window ();
  33. Lisp_Object Fset_window_buffer (), Fsplit_window (), Frecenter ();
  34.  
  35. static void replace_window (), unshow_buffer ();
  36. static int save_window_save ();
  37.  
  38. extern int minibuf_prompt_width;
  39.  
  40. /* This is the window which displays the minibuffer.
  41. It is always the same window.  */
  42.  
  43. Lisp_Object minibuf_window;
  44.  
  45. /* This is the window in which the terminal's cursor should
  46.  be left when nothing is being done with it.  This must
  47.  always be a leaf window, and its buffer is selected by
  48.  the top level editing loop at the end of each command.  */
  49.  
  50. Lisp_Object selected_window;
  51.  
  52. /* Non-nil means it is the window for C-M-v to scroll
  53.    when the minibuffer is selected.  */
  54.  
  55. Lisp_Object Vminibuf_scroll_window;
  56.  
  57. /* Non-nil means it's function to call to display temp buffers.  */
  58.  
  59. Lisp_Object Vtemp_buffer_show_hook;
  60.  
  61. /* If a window gets smaller than either of these, it is removed. */
  62.  
  63. int window_min_height;
  64. int window_min_width;
  65.  
  66. /* Nonzero implies pop_to_buffer should create windows. */
  67.  
  68. int pop_up_windows;
  69.  
  70. /* display-buffer always splits the largest window 
  71.  if that window is more than this high */
  72.  
  73. int split_height_threshold;
  74.  
  75. /* Number of lines of continuity in scrolling by screenfuls.  */
  76.  
  77. int next_screen_context_lines;
  78.  
  79. /* Incremented for each window created.  */
  80.  
  81. static int sequence_number;
  82.  
  83. DEFUN ("windowp", Fwindowp, Swindowp, 1, 1, 0,
  84.   "Returns t if OBJ is a window.")
  85.   (obj)
  86.      Lisp_Object obj;
  87. {
  88.   return XTYPE (obj) == Lisp_Window ? Qt : Qnil;
  89. }
  90.  
  91. static Lisp_Object
  92. make_window ()
  93. {
  94.   register Lisp_Object val;
  95.   register struct window *p;
  96.  
  97.   /* Add sizeof (Lisp_Object) here because sizeof (struct Lisp_Vector)
  98.      includes the first element.  */
  99.   val = Fmake_vector (
  100.     make_number ((sizeof (struct window) - sizeof (struct Lisp_Vector)
  101.           + sizeof (Lisp_Object))
  102.          / sizeof (Lisp_Object)),
  103.     Qnil);
  104.   XSETTYPE (val, Lisp_Window);
  105.   p = XWINDOW (val);
  106.   XFASTINT (p->sequence_number) = ++sequence_number;
  107.   XFASTINT (p->left) = XFASTINT (p->top)
  108.     = XFASTINT (p->height) = XFASTINT (p->width)
  109.       = XFASTINT (p->hscroll) = 0;
  110.   XFASTINT (p->last_point_x) = XFASTINT (p->last_point_y) = 0;
  111.   p->start = Fmake_marker ();
  112.   p->pointm = Fmake_marker ();
  113.   XFASTINT (p->use_time) = 0;
  114.   return val;
  115. }
  116.  
  117. DEFUN ("selected-window", Fselected_window, Sselected_window, 0, 0, 0,
  118.   "Return the window that the cursor now appears in and commands apply to.")
  119.   ()
  120. {
  121.   return selected_window;
  122. }
  123.  
  124. DEFUN ("minibuffer-window", Fminibuffer_window, Sminibuffer_window, 0, 0, 0,
  125.   "Return the window used for minibuffers.")
  126.   ()
  127. {
  128.   return minibuf_window;
  129. }
  130.  
  131. DEFUN ("pos-visible-in-window-p", Fpos_visible_in_window_p,
  132.   Spos_visible_in_window_p, 0, 2, 0,
  133.   "Return t if position POS is currently on the screen in WINDOW.\n\
  134. Returns nil if that position is scrolled vertically out of view.\n\
  135. POS defaults to point; WINDOW, to the selected window.")
  136.   (pos, window)
  137.      Lisp_Object pos, window;
  138. {
  139.   register struct window *w;
  140.   register int top;
  141.   register int height;
  142.   register int posint;
  143.   register struct buffer *buf;
  144.   struct position posval;
  145.  
  146.   if (NULL (pos))
  147.     posint = point;
  148.   else
  149.     {
  150.       CHECK_NUMBER_COERCE_MARKER (pos, 0);
  151.       posint = XINT (pos);
  152.     }
  153.  
  154.   if (NULL (window))
  155.     window = selected_window;
  156.   else
  157.     CHECK_WINDOW (window, 1);
  158.   w = XWINDOW (window);
  159.   top = marker_position (w->start);
  160.  
  161.   if (posint < top)
  162.     return Qnil;
  163.  
  164.   height = XFASTINT (w->height) - !EQ (window, minibuf_window);
  165.  
  166.   buf = XBUFFER (w->buffer);
  167.   if (XFASTINT (w->last_modified) >= BUF_MODIFF (buf))
  168.     {
  169.       /* If screen is up to date,
  170.      use the info recorded about how much text fit on it. */
  171.       if (posint < BUF_Z (buf) - XFASTINT (w->window_end_pos)
  172.       || (XFASTINT (w->window_end_vpos) < height))
  173.     return Qt;
  174.       return Qnil;
  175.     }
  176.   else
  177.     {
  178.       if (posint > BUF_Z (buf))
  179.     return Qnil;
  180.       /* If that info is not correct, calculate afresh */
  181.       posval = *compute_motion (top, 0, 0,
  182.                    posint, height, 0,
  183.                    XFASTINT (w->width) - 1
  184.                    - (XFASTINT (w->width) + XFASTINT (w->left) != XFASTINT (XWINDOW (minibuf_window)->width)),
  185.  
  186.                    XINT (w->hscroll), 0);
  187.       return posval.vpos < height ? Qt : Qnil;
  188.     }
  189. }
  190.  
  191. static struct window *
  192. decode_window (window)
  193.      register Lisp_Object window;
  194. {
  195.   if (NULL (window))
  196.     return XWINDOW (selected_window);
  197.  
  198.   CHECK_WINDOW (window, 0);
  199.   return XWINDOW (window);
  200. }
  201.  
  202. DEFUN ("window-buffer", Fwindow_buffer, Swindow_buffer, 0, 1, 0,
  203.   "Return the buffer that WINDOW is displaying.")
  204.   (window)
  205.      Lisp_Object window;
  206. {
  207.   return decode_window (window)->buffer;
  208. }
  209.  
  210. DEFUN ("window-height", Fwindow_height, Swindow_height, 0, 1, 0,
  211.   "Return the number of lines in WINDOW (including its mode line).")
  212.   (window)
  213.      Lisp_Object window;
  214. {
  215.   return decode_window (window)->height;
  216. }
  217.  
  218. DEFUN ("window-width", Fwindow_width, Swindow_width, 0, 1, 0,
  219.   "Return the number of columns in WINDOW.")
  220.   (window)
  221.      Lisp_Object window;
  222. {
  223.   register int w = decode_window (window)->width;
  224.   /* If this window does not end at the right margin,
  225.      must deduct one column for the border */
  226.   if (w + decode_window (window)->left == screen_width)
  227.     return w;
  228.   return w - 1;
  229. }
  230.  
  231. DEFUN ("window-hscroll", Fwindow_hscroll, Swindow_hscroll, 0, 1, 0,
  232.   "Return the number of columns by which WINDOW is scrolled from left margin.")
  233.   (window)
  234.      Lisp_Object window;
  235. {
  236.   return decode_window (window)->hscroll;
  237. }
  238.  
  239. DEFUN ("set-window-hscroll", Fset_window_hscroll, Sset_window_hscroll, 2, 2, 0,
  240.   "Set number of columns WINDOW is scrolled from left margin to NCOL.\n\
  241. NCOL should be zero or positive.")
  242.   (window, ncol)
  243.      register Lisp_Object window, ncol;
  244. {
  245.   register struct window *w;
  246.  
  247.   CHECK_NUMBER (ncol, 1);
  248.   if (XINT (ncol) < 0) XFASTINT (ncol) = 0;
  249.   if (XFASTINT (ncol) >= (1 << (SHORTBITS - 1)))
  250.     args_out_of_range (ncol, Qnil);
  251.   w = decode_window (window);
  252.   if (w->hscroll != ncol)
  253.     clip_changed = 1;        /* Prevent redisplay shortcuts */
  254.   w->hscroll = ncol;
  255.   return ncol;
  256. }
  257.  
  258. DEFUN ("window-edges", Fwindow_edges, Swindow_edges, 0, 1, 0,
  259.   "Return a list of the edge coordinates of WINDOW.\n\
  260. \(LEFT TOP RIGHT BOTTOM), all relative to 0, 0 at top left corner of screen.\n\
  261. RIGHT is one more than the rightmost column used by WINDOW,\n\
  262. and BOTTOM is one more than the bottommost row used by WINDOW\n\
  263.  and its mode-line.")
  264.   (window)
  265.      Lisp_Object window;
  266. {
  267.   register struct window *w = decode_window (window);
  268.  
  269.   return Fcons (w->left, Fcons (w->top,
  270.            Fcons (make_number (XFASTINT (w->left) + XFASTINT (w->width)),
  271.           Fcons (make_number (XFASTINT (w->top)
  272.                       + XFASTINT (w->height)),
  273.              Qnil))));
  274. }
  275.  
  276. DEFUN ("window-point", Fwindow_point, Swindow_point, 0, 1, 0,
  277.   "Return current value of point in WINDOW.\n\
  278. For a nonselected window, this is the value point would have\n\
  279. if that window were selected.\n\
  280. \n\
  281. Note that, when WINDOW is the selected window and its buffer\n\
  282. is also currently selected, the value returned is the same as (point).\n\
  283. It would be more strictly correct to return the `top-level' value\n\
  284. of point, outside of any  save-excursion  forms.\n\
  285. But that is hard to define.")
  286.   (window)
  287.      Lisp_Object window;
  288. {
  289.   register struct window *w = decode_window (window);
  290.  
  291.   if (w == XWINDOW (selected_window)
  292.       && current_buffer == XBUFFER (w->buffer))
  293.     return Fpoint ();
  294.   return Fmarker_position (w->pointm);
  295. }
  296.  
  297. DEFUN ("window-start", Fwindow_start, Swindow_start, 0, 1, 0,
  298.   "Return position at which display currently starts in WINDOW.")
  299.   (window)
  300.      Lisp_Object window;
  301. {
  302.   return Fmarker_position (decode_window (window)->start);
  303. }
  304.  
  305. DEFUN ("set-window-point", Fset_window_point, Sset_window_point, 2, 2, 0,
  306.   "Make point value in WINDOW be at position POS in WINDOW's buffer.")
  307.   (window, pos)
  308.      Lisp_Object window, pos;
  309. {
  310.   register struct window *w = decode_window (window);
  311.  
  312.   CHECK_NUMBER_COERCE_MARKER (pos, 1);
  313.   if (w == XWINDOW (selected_window))
  314.     Fgoto_char (pos);
  315.   else
  316.     set_marker_restricted (w->pointm, pos, w->buffer);
  317.   return pos;
  318. }
  319.  
  320. DEFUN ("set-window-start", Fset_window_start, Sset_window_start, 2, 3, 0,
  321.   "Make display in WINDOW start at position POS in WINDOW's buffer.\n\
  322. Optional third arg NOFORCE non-nil inhibits next redisplay\n\
  323. from overriding motion of point in order to display at this exact start.")
  324.   (window, pos, noforce)
  325.      Lisp_Object window, pos, noforce;
  326. {
  327.   register struct window *w = decode_window (window);
  328.  
  329.   CHECK_NUMBER_COERCE_MARKER (pos, 1);
  330.   set_marker_restricted (w->start, pos, w->buffer);
  331.   /* this is not right, but much easier than doing what is right.  */
  332.   w->start_at_line_beg = Qnil;
  333.   if (NULL (noforce))
  334.     w->force_start = Qt;
  335.   w->update_mode_line = Qt;
  336.   XFASTINT (w->last_modified) = 0;
  337.   return pos;
  338. }
  339.  
  340. DEFUN ("delete-window", Fdelete_window, Sdelete_window, 0, 1, "",
  341.   "Remove WINDOW from the display.  Default is selected window.")
  342.   (window)
  343.      register Lisp_Object window;
  344. {
  345.   int osize;
  346.   register Lisp_Object tem, parent;
  347.   register struct window *p;
  348.   register struct window *par;
  349.  
  350.   if (NULL (window))
  351.     window = selected_window;
  352.   else
  353.     CHECK_WINDOW (window, 0);
  354.  
  355.   p = XWINDOW (window);
  356.   parent = p->parent;
  357.   if (NULL (parent))
  358.     error ("Attempt to delete minibuffer or sole ordinary window");
  359.   par=XWINDOW (parent);
  360.  
  361.   windows_or_buffers_changed++;
  362.  
  363.   if (EQ (window, selected_window))
  364.     Fselect_window (Fnext_window (window, Qnil));
  365.  
  366.   tem = p->buffer;
  367.   /* tem is null for dummy parent windows
  368.      (which have inferiors but not any contents themselves) */
  369.   if (!NULL (tem))
  370.     {
  371.       unshow_buffer (p);
  372.       unchain_marker (p->pointm);
  373.       unchain_marker (p->start);
  374.     }
  375.  
  376.   tem = p->next;
  377.   if (!NULL (tem))
  378.     XWINDOW (tem)->prev = p->prev;
  379.  
  380.   tem = p->prev;
  381.   if (!NULL (tem))
  382.     XWINDOW (tem)->next = p->next;
  383.  
  384.   if (EQ (window, par->hchild))
  385.     par->hchild = p->next;
  386.   if (EQ (window, par->vchild))
  387.     par->vchild = p->next;
  388.  
  389.   /* Stretch the siblings to use all the available space */
  390.   if (!NULL (par->vchild))
  391.     {
  392.       /* It's a vertical combination */
  393.       osize = XFASTINT (par->height);
  394.       XFASTINT (par->height)
  395.     -= XFASTINT (p->height);
  396.       set_window_height (parent, osize, 1);
  397.     }
  398.   if (!NULL (par->hchild))
  399.     {
  400.       /* It's a horizontal combination */
  401.       osize = XFASTINT (par->width);
  402.       XFASTINT (par->width)
  403.     -= XFASTINT (p->width);
  404.       set_window_width (parent, osize, 1);
  405.     }
  406.  
  407.   /* If parent now has only one child,
  408.      put the child into the parent's place.  */
  409.  
  410.   tem = par->hchild;
  411.   if (NULL (tem))
  412.     tem = par->vchild;
  413.   if (NULL (XWINDOW (tem)->next))
  414.     replace_window (parent, tem);
  415.   return Qnil;
  416. }
  417.  
  418. /* Put replacement into the window structure in place of old. */
  419. static void
  420. replace_window (old, replacement)
  421.      Lisp_Object old, replacement;
  422. {
  423.   register Lisp_Object tem;
  424.   register struct window *o = XWINDOW (old), *p = XWINDOW (replacement);
  425.  
  426.   p->left = o->left;
  427.   p->top = o->top;
  428.   p->width = o->width;
  429.   p->height = o->height;
  430.  
  431.   p->next = tem = o->next;
  432.   if (!NULL (tem))
  433.     XWINDOW (tem)->prev = replacement;
  434.  
  435.   p->prev = tem = o->prev;
  436.   if (!NULL (tem))
  437.     XWINDOW (tem)->next = replacement;
  438.  
  439.   p->parent = tem = o->parent;
  440.   if (!NULL (tem))
  441.     {
  442.       if (EQ (XWINDOW (tem)->vchild, old))
  443.     XWINDOW (tem)->vchild = replacement;
  444.       if (EQ (XWINDOW (tem)->hchild, old))
  445.     XWINDOW (tem)->hchild = replacement;
  446.     }
  447.  
  448. /*** Here, if replacement is a vertical combination
  449. and so is its new parent, we should make replacement's
  450. children be children of that parent instead.  ***/
  451. }
  452.  
  453. DEFUN ("next-window", Fnext_window, Snext_window, 0, 2, 0,
  454.   "Return next window after WINDOW in canonical ordering of windows.\n\
  455. Optional second arg MINIBUF t means count the minibuffer window\n\
  456. even if not active.  If MINIBUF is neither t nor nil it means\n\
  457. not to count the minibuffer even if it is active.")
  458.   (window, mini)
  459.      register Lisp_Object window, mini;
  460. {
  461.   register Lisp_Object tem;
  462.   if (NULL (window))
  463.     window = selected_window;
  464.   else
  465.     CHECK_WINDOW (window, 0);
  466.   do
  467.     {
  468.       while (tem = XWINDOW (window)->next, NULL (tem))
  469.     if (tem = XWINDOW (window)->parent, !NULL (tem))
  470.       window = tem;
  471.         else  /* window must be minibuf_window now */
  472.       {
  473.         tem = XWINDOW (window)->prev;
  474.         break;
  475.       }
  476.       window = tem;
  477.       while (1)
  478.     {
  479.       if (!NULL (XWINDOW (window)->hchild))
  480.         window = XWINDOW (window)->hchild;
  481.       else if (!NULL (XWINDOW (window)->vchild))
  482.         window = XWINDOW (window)->vchild;
  483.       else break;
  484.     }
  485.     }
  486.   while (EQ (window, minibuf_window) && !EQ (mini, Qt)
  487.      && (!NULL (mini) || minibuf_level == 0));
  488.   return window;
  489. }
  490.  
  491. DEFUN ("previous-window", Fprevious_window, Sprevious_window, 0, 1, 0,
  492.   "Return previous window before WINDOW in canonical ordering of windows.")
  493.   (window)
  494.      register Lisp_Object window;
  495. {
  496.   register Lisp_Object tem;
  497.   if (NULL (window))
  498.     window = selected_window;
  499.   else
  500.     CHECK_WINDOW (window, 0);
  501.   do  /* at least once, and until not the minibuffer */
  502.     {
  503.       while (tem = XWINDOW (window)->prev, NULL (tem))
  504.     if (tem = XWINDOW (window)->parent, !NULL (tem))
  505.       window = tem;
  506.         else  /* window must be the root window now */
  507.       {
  508.         tem = minibuf_window;
  509.         break;
  510.       }
  511.       window = tem;
  512.       while (1)
  513.     {
  514.       if (!NULL (XWINDOW (window)->hchild))
  515.         window = XWINDOW (window)->hchild;
  516.       else if (!NULL (XWINDOW (window)->vchild))
  517.         window = XWINDOW (window)->vchild;
  518.       else break;
  519.       while (tem = XWINDOW (window)->next, !NULL (tem))
  520.         window = tem;
  521.     }
  522.     }
  523.   while (EQ (window, minibuf_window) && minibuf_level == 0);
  524.   return window;
  525. }
  526.  
  527. DEFUN ("other-window", Fother_window, Sother_window, 1, 1, "p",
  528.   "Select the ARG'th different window.")
  529.   (n)
  530.      register Lisp_Object n;
  531. {
  532.   register int i;
  533.   register Lisp_Object w;
  534.  
  535.   CHECK_NUMBER (n, 0);
  536.   w = selected_window;
  537.   i = XINT (n);
  538.  
  539.   while (i > 0)
  540.     {
  541.       w = Fnext_window (w, Qnil);
  542.       i--;
  543.     }
  544.   while (i < 0)
  545.     {
  546.       w = Fprevious_window (w);
  547.       i++;
  548.     }
  549.   Fselect_window (w);
  550.   return Qnil;
  551. }
  552.  
  553. static Lisp_Object
  554. window_loop (type, obj)
  555.      int type;
  556.      register Lisp_Object obj;
  557. {
  558.   register Lisp_Object w, tem, ret_w;
  559.   Lisp_Object w1, start_w;
  560.   register struct window *p, *q;
  561.  
  562.   w = minibuf_window;
  563.   ret_w = Qnil;
  564.   while (1)
  565.     {
  566.       p = XWINDOW (w);
  567.       w1 = Fnext_window (w, Qt);
  568.       if (!EQ (w, minibuf_window))
  569.     switch (type)
  570.       {
  571.       case 1:
  572.         if (XBUFFER (p->buffer) == XBUFFER (obj))
  573.           return w;
  574.         break;
  575.  
  576.       case 2:
  577.         /* t as arg means consider only full-width windows */
  578.         if (!NULL (obj) && XFASTINT (p->width) != screen_width)
  579.           break;
  580.         if (NULL (ret_w) ||
  581.         XFASTINT (XWINDOW (ret_w)->use_time) > XFASTINT (p->use_time))
  582.           ret_w = w;
  583.         break;
  584.  
  585.       case 3:
  586.         if (p != XWINDOW (obj))
  587.           Fdelete_window (w);
  588.         break;
  589.  
  590.       case 4:
  591.         if (EQ (p->buffer, obj))
  592.           {
  593.         if (NULL (p->parent))
  594.           {
  595.             tem = Fother_buffer (obj);
  596.             if (NULL (tem))
  597.               tem = Fget_buffer_create (build_string ("*scratch*"));
  598.             Fset_window_buffer (w, tem);
  599.             Fset_buffer (p->buffer);
  600.           }
  601.         else
  602.           Fdelete_window (w);
  603.           }
  604.         break;
  605.  
  606.       case 5:
  607.         q = XWINDOW (ret_w);
  608.         if (NULL (ret_w) ||
  609.         (XFASTINT (p->height) * XFASTINT (p->width))
  610.         >
  611.         (XFASTINT (q->height) * XFASTINT (q->width)))
  612.           ret_w = w;
  613.         break;
  614.  
  615.       case 6:
  616.         if (EQ (p->buffer, obj))
  617.           {
  618.         tem = Fother_buffer (obj);
  619.         if (NULL (tem))
  620.           tem = Fget_buffer_create (build_string ("*scratch*"));
  621.         Fset_window_buffer (w, tem);
  622.           }
  623.         break;
  624.       }
  625.       w = w1;
  626.       if (EQ (w, minibuf_window))
  627.     return ret_w;
  628.     }
  629. }     
  630.  
  631. DEFUN ("get-lru-window", Fget_lru_window, Sget_lru_window, 0, 0, 0,
  632.   "Return the window least recently selected or used for display.")
  633.   ()
  634. {
  635.   register Lisp_Object w;
  636.   /* First try for a window that is full-width */
  637.   w = window_loop (2, Qt);
  638.   if (!NULL (w) && !EQ (w, selected_window))
  639.     return w;
  640.   /* If none of them, try the rest */
  641.   return window_loop (2, Qnil);
  642. }
  643.  
  644. DEFUN ("get-largest-window", Fget_largest_window, Sget_largest_window, 0, 0, 0,
  645.   "Return the largest window in area.")
  646.   ()
  647. {
  648.   return window_loop (5, Qnil);
  649. }
  650.  
  651. DEFUN ("get-buffer-window", Fget_buffer_window, Sget_buffer_window, 1, 1, 0,
  652.   "Return a window currently displaying BUFFER, or nil if none.")
  653.   (buffer)
  654.      Lisp_Object buffer;
  655. {
  656.   buffer = Fget_buffer (buffer);
  657.   if (XTYPE (buffer) == Lisp_Buffer)
  658.     return window_loop (1, buffer);
  659.   else return Qnil;
  660. }
  661.  
  662. DEFUN ("delete-other-windows", Fdelete_other_windows, Sdelete_other_windows,
  663.   0, 1, "",
  664.   "Make WINDOW (or the selected window) fill the screen.")
  665.   (w)
  666.      Lisp_Object w;
  667. {
  668.   window_loop (3, !NULL (w) ? w : selected_window);
  669.   return Qnil;
  670. }
  671.  
  672. DEFUN ("delete-windows-on", Fdelete_windows_on, Sdelete_windows_on,
  673.   1, 1, "bDelete windows on (buffer): ",
  674.   "Delete all windows showing BUFFER.")
  675.   (buffer)
  676.      Lisp_Object buffer;
  677. {
  678.   if (!NULL (buffer))
  679.     {
  680.       buffer = Fget_buffer (buffer);
  681.       CHECK_BUFFER (buffer, 0);
  682.       window_loop (4, buffer);
  683.     }
  684.   return Qnil;
  685. }
  686.  
  687. DEFUN ("replace-buffer-in-windows", Freplace_buffer_in_windows,
  688.   Sreplace_buffer_in_windows,
  689.   1, 1, "bReplace buffer in windows: ",
  690.   "Replace BUFFER with some other buffer in all windows showing it.")
  691.   (buffer)
  692.      Lisp_Object buffer;
  693. {
  694.   if (!NULL (buffer))
  695.     {
  696.       buffer = Fget_buffer (buffer);
  697.       CHECK_BUFFER (buffer, 0);
  698.       window_loop (6, buffer);
  699.     }
  700.   return Qnil;
  701. }
  702.  
  703. /* Set the height of WINDOW and all its inferiors.  */
  704. /* Normally the window is deleted if it gets too small.
  705.    nodelete nonzero means do not do this.
  706.    (The caller should check later and do so if appropriate)  */
  707.  
  708. set_window_height (window, height, nodelete)
  709.      Lisp_Object window;
  710.      int height;
  711.      int nodelete;
  712. {
  713.   register struct window *w = XWINDOW (window);
  714.   register struct window *c;
  715.   int oheight = XFASTINT (w->height);
  716.   int top, pos, lastbot, opos, lastobot;
  717.   Lisp_Object child;
  718.  
  719.   if (window_min_height < 2)
  720.     window_min_height = 2;
  721.  
  722.   if (!nodelete
  723.       && ! NULL (w->parent)
  724.       && height < (EQ(window, minibuf_window) ? 1 : window_min_height))
  725.     {
  726.       Fdelete_window (window);
  727.       return;
  728.     }
  729.  
  730.   XFASTINT (w->last_modified) = 0;
  731.   windows_or_buffers_changed++;
  732.   XFASTINT (w->height) = height;
  733.   if (!NULL (w->hchild))
  734.     {
  735.       for (child = w->hchild; !NULL (child); child = XWINDOW (child)->next)
  736.     {
  737.       XWINDOW (child)->top = w->top;
  738.       set_window_height (child, height, nodelete);
  739.     }
  740.     }
  741.   else if (!NULL (w->vchild))
  742.     {
  743.       lastbot = top = XFASTINT (w->top);
  744.       lastobot = 0;
  745.       for (child = w->vchild; !NULL (child); child = c->next)
  746.     {
  747.       c = XWINDOW (child);
  748.  
  749.       opos = lastobot + XFASTINT (c->height);
  750.  
  751.       XFASTINT (c->top) = lastbot;
  752.  
  753.       pos = (((opos * height) << 1) + oheight) / (oheight << 1);
  754.  
  755.       /* Avoid confusion: inhibit deletion of child if becomes too small */
  756.       set_window_height (child, pos + top - lastbot, 1);
  757.  
  758.       /* Now advance child to next window,
  759.          and set lastbot if child was not just deleted.  */
  760.       lastbot = pos + top, lastobot = opos;
  761.     }
  762.       /* Now delete any children that became too small.  */
  763.       if (!nodelete)
  764.     for (child = w->vchild; !NULL (child); child = XWINDOW (child)->next)
  765.       {
  766.         set_window_height (child, XINT (XWINDOW (child)->height), 0);
  767.       }
  768.     }
  769. }
  770.  
  771. /* Recursively set width of WINDOW and its inferiors. */
  772.  
  773. set_window_width (window, width, nodelete)
  774.      Lisp_Object window;
  775.      int width;
  776.      int nodelete;
  777. {
  778.   register struct window *w = XWINDOW (window);
  779.   register struct window *c;
  780.   int owidth = XFASTINT (w->width);
  781.   int left, pos, lastright, opos, lastoright;
  782.   Lisp_Object child;
  783.  
  784.   if (!nodelete
  785.       && ! NULL (w->parent)
  786.       && width < window_min_width)
  787.     {
  788.       Fdelete_window (window);
  789.       return;
  790.     }
  791.  
  792.   XFASTINT (w->last_modified) = 0;
  793.   windows_or_buffers_changed++;
  794.   XFASTINT (w->width) = width;
  795.   if (!NULL (w->vchild))
  796.     {
  797.       for (child = w->vchild; !NULL (child); child = XWINDOW (child)->next)
  798.     {
  799.       XWINDOW (child)->left = w->left;
  800.       set_window_width (child, width, nodelete);
  801.     }
  802.     }
  803.   else if (!NULL (w->hchild))
  804.     {
  805.       lastright = left = XFASTINT (w->left);
  806.       lastoright = 0;
  807.       for (child = w->hchild; !NULL (child); child = c->next)
  808.     {
  809.       c = XWINDOW (child);
  810.  
  811.       opos = lastoright + XFASTINT (c->width);
  812.  
  813.       XFASTINT (c->left) = lastright;
  814.  
  815.       pos = (((opos * width) << 1) + owidth) / (owidth << 1);
  816.  
  817.       /* Inhibit deletion for becoming too small */
  818.       set_window_width (child, pos + left - lastright, 1);
  819.  
  820.       /* Now advance child to next window,
  821.          and set lastright if child was not just deleted.  */
  822.       lastright = pos + left, lastoright = opos;
  823.     }
  824.       /* Delete children that became too small */
  825.       if (!nodelete)
  826.     for (child = w->hchild; !NULL (child); child = XWINDOW (child)->next)
  827.       {
  828.         set_window_width (child, XINT (XWINDOW (child)->width), 0);
  829.       }
  830.     }
  831. }
  832.  
  833. static int window_select_count;
  834.  
  835. DEFUN ("set-window-buffer", Fset_window_buffer, Sset_window_buffer, 2, 2, 0,
  836.   "Make WINDOW display BUFFER as its contents.\n\
  837. BUFFER can be a buffer or buffer name.")
  838.   (window, buffer)
  839.      register Lisp_Object window, buffer;
  840. {
  841.   register Lisp_Object tem;
  842.   register struct window *w = decode_window (window);
  843.  
  844.   buffer = Fget_buffer (buffer);
  845.   CHECK_BUFFER (buffer, 1);
  846.  
  847.   if (NULL (XBUFFER (buffer)->name))
  848.     error ("Attempt to display deleted buffer");
  849.  
  850.   tem = w->buffer;
  851.   if (!NULL (tem))
  852.     unshow_buffer (w);
  853.  
  854.   w->buffer = buffer;
  855.   Fset_marker (w->pointm,
  856.            make_number (BUF_PT (XBUFFER (buffer))),
  857.            buffer);
  858.   set_marker_restricted (w->start,
  859.              make_number (XBUFFER (buffer)->last_window_start),
  860.              buffer);
  861.   w->start_at_line_beg = Qnil;
  862.   XFASTINT (w->last_modified) = 0;
  863.   windows_or_buffers_changed++;
  864.   if (EQ (window, selected_window))
  865.     Fset_buffer (buffer);
  866.  
  867.   return Qnil;
  868. }
  869.  
  870. /* Record info on buffer window w is displaying
  871.    when it is about to cease to display that buffer.  */
  872. static void
  873. unshow_buffer (w)
  874.      register struct window *w;
  875. {
  876.   register Lisp_Object buf;
  877.   buf = w->buffer;
  878.  
  879.   if (XBUFFER (buf) != XMARKER (w->pointm)->buffer)
  880.     abort ();
  881.  
  882.   if (w == XWINDOW (selected_window)
  883.       || ! EQ (buf, XWINDOW (selected_window)->buffer))
  884.     /* Do this except when the selected window's buffer
  885.        is being removed from some other window.  */
  886.     XBUFFER (buf)->last_window_start = marker_position (w->start);
  887.  
  888.   /* Point in the selected window's buffer
  889.      is actually stored in that buffer, and the window's pointm isn't used.
  890.      So don't clobber point in that buffer.  */
  891.   if (! EQ (buf, XWINDOW (selected_window)->buffer))
  892.     BUF_PT (XBUFFER (buf))
  893.       = clip_to_bounds (BUF_BEGV (XBUFFER (buf)),
  894.             marker_position (w->pointm),
  895.             BUF_ZV (XBUFFER (buf)));
  896. }
  897.  
  898. DEFUN ("select-window", Fselect_window, Sselect_window, 1, 1, 0,
  899.   "Select WINDOW.  Most editing will apply to WINDOW's buffer.\n\
  900. The main editor command loop selects the buffer of the selected window\n\
  901. before each command.")
  902.   (window)
  903.      register Lisp_Object window;
  904. {
  905.   register struct window *w;
  906.   register struct window *ow = XWINDOW (selected_window);
  907.  
  908.   CHECK_WINDOW (window, 0);
  909.  
  910.   w = XWINDOW (window);
  911.  
  912.   if (NULL (w->buffer))
  913.     error ("Trying to select window with no buffer");
  914.  
  915.   XFASTINT (w->use_time) = ++window_select_count;
  916.   if (EQ (window, selected_window))
  917.     return window;
  918.  
  919.   Fset_marker (ow->pointm, make_number (BUF_PT (XBUFFER (ow->buffer))),
  920.            ow->buffer);
  921.  
  922.   selected_window = window;
  923.  
  924.   record_buffer (w->buffer);
  925.   Fset_buffer (w->buffer);
  926.  
  927.   /* Go to the point recorded in the window.
  928.      This is important when the buffer is in more
  929.      than one window.  It also matters when
  930.      redisplay_window has altered point after scrolling,
  931.      because it makes the change only in the window.  */
  932.   SET_PT (marker_position (w->pointm));
  933.   if (point < BEGV)
  934.     point = BEGV;
  935.   if (point > ZV)
  936.     point = ZV;
  937.  
  938.   windows_or_buffers_changed++;
  939.  
  940.   return window;
  941. }
  942.  
  943. DEFUN ("display-buffer", Fdisplay_buffer, Sdisplay_buffer, 1, 2, 0,
  944.   "Make BUFFER appear in some window but don't select it.\n\
  945. BUFFER can be a buffer or a buffer name.\n\
  946. If BUFFER is shown already in some window, just uses that one,\n\
  947. unless the window is the selected window and NOTTHISWINDOW is non-nil.\n\
  948. Returns the window displaying BUFFER.")
  949.   (buffer, notthiswindow)
  950.      register Lisp_Object buffer, notthiswindow;
  951. {
  952.   register Lisp_Object window;
  953.  
  954.   buffer = Fget_buffer (buffer);
  955.   CHECK_BUFFER (buffer, 0);
  956.  
  957.   if (NULL (notthiswindow)
  958.       && XBUFFER (XWINDOW (selected_window)->buffer) == XBUFFER (buffer))
  959.     return selected_window;
  960.  
  961.   window = Fget_buffer_window (buffer);
  962.   if (!NULL (window)
  963.       && (NULL (notthiswindow) || !EQ (window, selected_window)))
  964.     return window;
  965.  
  966.   if (pop_up_windows)
  967.     {
  968.       /* Don't try to create a window if would get an error */
  969.       if (window_min_height < 2)
  970.     window_min_height = 2;
  971.       if (split_height_threshold < window_min_height << 1)
  972.     split_height_threshold = window_min_height << 1;
  973.  
  974.       window = Fget_largest_window ();
  975.       if (window_height (window) >= split_height_threshold
  976.       &&
  977.       XFASTINT (XWINDOW (window)->width) == screen_width)
  978.     window = Fsplit_window (window, Qnil, Qnil);
  979.       else
  980.     {
  981.       window = Fget_lru_window ();
  982.       if ((EQ (window, selected_window)
  983.            || (EQ (selected_window, minibuf_window)
  984.            && EQ (window, XWINDOW (minibuf_window)->prev)))
  985.           && window_height (window) >= window_min_height << 1)
  986.         window = Fsplit_window (window, Qnil, Qnil);
  987.     }
  988.     }
  989.   else
  990.     window = Fget_lru_window ();
  991.  
  992.   Fset_window_buffer (window, buffer);
  993.   return window;
  994. }
  995.  
  996. temp_output_buffer_show (buf)
  997.      register Lisp_Object buf;
  998. {
  999.   register struct buffer *old = current_buffer;
  1000.   register Lisp_Object window;
  1001.   register struct window *w;
  1002.  
  1003.   Fset_buffer (buf);
  1004.   XBUFFER (buf)->save_modified = MODIFF;
  1005.   BEGV = BEG;
  1006.   ZV = Z;
  1007.   SET_PT (BEG);
  1008.   clip_changed = 1;
  1009.   set_buffer_internal (old);
  1010.  
  1011.   if (!EQ (Vtemp_buffer_show_hook, Qnil))
  1012.     call1 (Vtemp_buffer_show_hook, buf);
  1013.   else
  1014.     {
  1015.       window = Fdisplay_buffer (buf, Qnil);
  1016.       Vminibuf_scroll_window = window;
  1017.       w = XWINDOW (window);
  1018.       XFASTINT (w->hscroll) = 0;
  1019.       set_marker_restricted (w->start, make_number (1), buf);
  1020.       set_marker_restricted (w->pointm, make_number (1), buf);
  1021.     }
  1022. }
  1023.  
  1024. static
  1025. make_dummy_parent (window)
  1026.      Lisp_Object window;
  1027. {
  1028.   register Lisp_Object old, new;
  1029.   register struct window *o, *p;
  1030.  
  1031.   old = window;
  1032.   XSETTYPE (old, Lisp_Vector);
  1033.   new = Fcopy_sequence (old);
  1034.   XSETTYPE (new, Lisp_Window);
  1035.  
  1036.   o = XWINDOW (old);
  1037.   p = XWINDOW (new);
  1038.   XFASTINT (p->sequence_number) = ++sequence_number;
  1039.  
  1040.   /* Put new into window structure in place of window */
  1041.   replace_window (window, new);
  1042.  
  1043.   o->next = Qnil;
  1044.   o->prev = Qnil;
  1045.   o->vchild = Qnil;
  1046.   o->hchild = Qnil;
  1047.   o->parent = new;
  1048.  
  1049.   p->start = Qnil;
  1050.   p->pointm = Qnil;
  1051.   p->buffer = Qnil;
  1052. }
  1053.  
  1054. DEFUN ("split-window", Fsplit_window, Ssplit_window, 0, 3, "",
  1055.   "Split WINDOW, putting SIZE lines in the first of the pair.\n\
  1056. WINDOW defaults to selected one and SIZE to half its size.\n\
  1057. If optional third arg HOR-FLAG is non-nil, split side by side\n\
  1058. and put SIZE columns in the first of the pair.")
  1059.   (window, chsize, horflag)
  1060.      Lisp_Object window, chsize, horflag;
  1061. {
  1062.   register Lisp_Object new;
  1063.   register struct window *o, *p;
  1064.   register int size;
  1065.  
  1066.   if (NULL (window))
  1067.     window = selected_window;
  1068.   else
  1069.     CHECK_WINDOW (window, 0);
  1070.  
  1071.   o = XWINDOW (window);
  1072.  
  1073.   if (NULL (chsize))
  1074.     {
  1075.       if (!NULL (horflag))
  1076.     /* Add 1 so we round up rather than down.
  1077.        This puts an excess column into the left-hand window,
  1078.        which is the one that certainly contains a border line.  */
  1079.     size = (1 + XFASTINT (o->width)) >> 1;
  1080.       else
  1081.     size = XFASTINT (o->height) >> 1;
  1082.     }
  1083.   else
  1084.     {
  1085.       CHECK_NUMBER (chsize, 1);
  1086.       size = XINT (chsize);
  1087.     }
  1088.  
  1089.   if (EQ (window, minibuf_window))
  1090.     error ("Attempt to split minibuffer window");
  1091.  
  1092.   if (NULL (horflag))
  1093.     {
  1094.       if (window_min_height < 2)
  1095.     window_min_height = 2;
  1096.  
  1097.       if (size < window_min_height ||
  1098.       size + window_min_height > XFASTINT (o->height))
  1099.     args_out_of_range_3 (window, chsize, horflag);
  1100.       if (NULL (o->parent) ||
  1101.       NULL (XWINDOW (o->parent)->vchild))
  1102.     {
  1103.       make_dummy_parent (window);
  1104.       new = o->parent;
  1105.       XWINDOW (new)->vchild = window;
  1106.     }
  1107.     }
  1108.   else
  1109.     {
  1110.       if (size < window_min_width ||
  1111.       size + window_min_width > XFASTINT (o->width))
  1112.     args_out_of_range_3 (window, chsize, horflag);
  1113.       if (NULL (o->parent) ||
  1114.       NULL (XWINDOW (o->parent)->hchild))
  1115.     {
  1116.       make_dummy_parent (window);
  1117.       new = o->parent;
  1118.       XWINDOW (new)->hchild = window;
  1119.     }
  1120.     }
  1121.  
  1122.   /* Now we know that window's parent is a vertical combination
  1123.      if we are dividing vertically, or a horizontal combination
  1124.      if we are making side-by-side windows */
  1125.  
  1126.   windows_or_buffers_changed++;
  1127.   new = make_window ();
  1128.   p = XWINDOW (new);
  1129.  
  1130.   p->next = o->next;
  1131.   if (!NULL (p->next))
  1132.     XWINDOW (p->next)->prev = new;
  1133.   p->prev = window;
  1134.   o->next = new;
  1135.   p->parent = o->parent;
  1136.  
  1137.   Fset_window_buffer (new, o->buffer);
  1138.  
  1139.   /* Apportion the available screen space among the two new windows */
  1140.  
  1141.   if (!NULL (horflag))
  1142.     {
  1143.       p->height = o->height;
  1144.       p->top = o->top;
  1145.       XFASTINT (p->width) = XFASTINT (o->width) - size;
  1146.       XFASTINT (o->width) = size;
  1147.       XFASTINT (p->left) = XFASTINT (o->left) + size;
  1148.     }
  1149.   else
  1150.     {
  1151.       p->left = o->left;
  1152.       p->width = o->width;
  1153.       XFASTINT (p->height) = XFASTINT (o->height) - size;
  1154.       XFASTINT (o->height) = size;
  1155.       XFASTINT (p->top) = XFASTINT (o->top) + size;
  1156.     }
  1157.  
  1158.   return new;
  1159. }
  1160.  
  1161. DEFUN ("enlarge-window", Fenlarge_window, Senlarge_window, 1, 2, "p",
  1162.   "Make current window ARG lines bigger.\n\
  1163. From program, optional second arg non-nil means grow sideways ARG columns.")
  1164.   (n, side)
  1165.      register Lisp_Object n, side;
  1166. {
  1167.   CHECK_NUMBER (n, 0);
  1168.   change_window_height (XINT (n), !NULL (side));
  1169.   return Qnil;
  1170. }
  1171.  
  1172. DEFUN ("shrink-window", Fshrink_window, Sshrink_window, 1, 2, "p",
  1173.   "Make current window ARG lines smaller.\n\
  1174. From program, optional second arg non-nil means shrink sideways ARG columns.")
  1175.   (n, side)
  1176.      register Lisp_Object n, side;
  1177. {
  1178.   CHECK_NUMBER (n, 0);
  1179.   change_window_height (-XINT (n), !NULL (side));
  1180.   return Qnil;
  1181. }
  1182.  
  1183. int
  1184. window_height (window)
  1185.      Lisp_Object window;
  1186. {
  1187.   register struct window *p = XWINDOW (window);
  1188.   return XFASTINT (p->height);
  1189. }
  1190.  
  1191. int
  1192. window_width (window)
  1193.      Lisp_Object window;
  1194. {
  1195.   register struct window *p = XWINDOW (window);
  1196.   return XFASTINT (p->width);
  1197. }
  1198.  
  1199. #define MINSIZE(window) \
  1200.   (widthflag ? window_min_width  \
  1201.    : (EQ (window, minibuf_window) ? 1 : window_min_height))
  1202.  
  1203. #define CURBEG(w) \
  1204.   *(widthflag ? (int *) &w->left : (int *) &w->top)
  1205.  
  1206. #define CURSIZE(w) \
  1207.   *(widthflag ? (int *) &w->width : (int *) &w->height)
  1208.  
  1209. /* Unlike set_window_height, this function
  1210.  also changes the heights of the siblings so as to
  1211.  keep everything consistent. */
  1212.  
  1213. change_window_height (delta, widthflag)
  1214.      register int delta;
  1215.      int widthflag;
  1216. {
  1217.   register Lisp_Object parent;
  1218.   Lisp_Object window;
  1219.   register struct window *p;
  1220.   int *sizep;
  1221.   int (*sizefun) () = widthflag ? window_width : window_height;
  1222.   register int (*setsizefun) () = widthflag ? set_window_width : set_window_height;
  1223.  
  1224.   if (window_min_height < 2)
  1225.     window_min_height = 2;
  1226.  
  1227.   window = selected_window;
  1228.   while (1)
  1229.     {
  1230.       p = XWINDOW (window);
  1231.       parent = p->parent;
  1232.       if (NULL (parent))
  1233.     {
  1234.       if (widthflag)
  1235.         error ("No other window to side of this one");
  1236.       break;
  1237.     }
  1238.       if (widthflag ? !NULL (XWINDOW (parent)->hchild)
  1239.       : !NULL (XWINDOW (parent)->vchild))
  1240.     break;
  1241.       window = parent;
  1242.     }
  1243.  
  1244.   sizep = &CURSIZE (p);
  1245.  
  1246.   if (*sizep + delta < MINSIZE (window))
  1247.     {
  1248.       Fdelete_window (window);
  1249.       return;
  1250.     }
  1251.  
  1252.   {
  1253.     register int maxdelta;
  1254.     register Lisp_Object tem;
  1255.  
  1256.     maxdelta = (!NULL (parent) ? (*sizefun) (parent) - *sizep
  1257.         : (tem = (!NULL (p->next) ? p->next : p->prev),
  1258.            (*sizefun) (tem) - MINSIZE (tem)));
  1259.  
  1260.     if (delta > maxdelta)
  1261.       /* This case traps trying to make the minibuffer
  1262.      the full screen, or make the only window aside from the
  1263.      minibuffer the full screen.  */
  1264.       delta = maxdelta;
  1265.   }
  1266.  
  1267.   if (!NULL (p->next) &&
  1268.       (*sizefun) (p->next) - delta >= MINSIZE (p->next))
  1269.     {
  1270.       (*setsizefun) (p->next, (*sizefun) (p->next) - delta, 0);
  1271.       (*setsizefun) (window, *sizep + delta, 0);
  1272.       CURBEG (XWINDOW (p->next)) += delta;
  1273.       /* This does not change size of p->next,
  1274.      but it propagates the new top edge to its children */
  1275.       (*setsizefun) (p->next, (*sizefun) (p->next), 0);
  1276.     }
  1277.   else if (!NULL (p->prev) &&
  1278.        (*sizefun) (p->prev) - delta >= MINSIZE (p->prev))
  1279.     {
  1280.       (*setsizefun) (p->prev, (*sizefun) (p->prev) - delta, 0);
  1281.       CURBEG (p) -= delta;
  1282.       (*setsizefun) (window, *sizep + delta, 0);
  1283.     }
  1284.   else
  1285.     {
  1286.       register int delta1;
  1287.       register int opht = (*sizefun) (parent);
  1288.  
  1289.       /* If trying to grow this window to or beyond size of the parent,
  1290.      make delta1 so big that, on shrinking back down,
  1291.      all the siblings end up with less than one line and are deleted.  */
  1292.       if (opht <= *sizep + delta)
  1293.     delta1 = opht * opht * 2;
  1294.       /* Otherwise, make delta1 just right so that if we add delta1
  1295.      lines to this window and to the parent, and then shrink
  1296.      the parent back to its original size, the new proportional
  1297.      size of this window will increase by delta.  */
  1298.       else
  1299.     delta1 = (delta * opht * 100) / ((opht - *sizep - delta) * 100);
  1300.  
  1301.       /* Add delta1 lines or columns to this window, and to the parent,
  1302.      keeping things consistent while not affecting siblings.  */
  1303.       CURSIZE (XWINDOW (parent)) = opht + delta1;
  1304.       (*setsizefun) (window, *sizep + delta1, 0);
  1305.  
  1306.       /* Squeeze out delta1 lines or columns from our parent,
  1307.      shriking this window and siblings proportionately.
  1308.      This brings parent back to correct size.
  1309.      Delta1 was calculated so this makes this window the desired size,
  1310.      taking it all out of the siblings.  */
  1311.       (*setsizefun) (parent, opht, 0);
  1312.     }
  1313.  
  1314.   XFASTINT (p->last_modified) = 0;
  1315. }
  1316. #undef MINSIZE
  1317. #undef CURBEG
  1318. #undef CURSIZE
  1319.  
  1320.  
  1321. /* Scroll window WINDOW (a Lisp object) by N lines.
  1322.    If NOERROR is 0, signal an error if that can't be done.
  1323.    If NOERROR is nonzero, return Qnil if successful
  1324.    and an error name otherwise.  */
  1325.  
  1326. static Lisp_Object
  1327. window_scroll (window, n, noerror)
  1328.      Lisp_Object window;
  1329.      int n;
  1330.      int noerror;
  1331. {
  1332.   register struct window *w = XWINDOW (window);
  1333.   register int opoint = point;
  1334.   register int ht, pos;
  1335.   register Lisp_Object tem;
  1336.   int lose;
  1337.   Lisp_Object bolp;
  1338.  
  1339.   ht = XFASTINT (w->height) - !EQ (window, minibuf_window);
  1340.  
  1341.   XFASTINT (tem) = point;
  1342.   tem = Fpos_visible_in_window_p (tem, window);
  1343.  
  1344.   if (NULL (tem))
  1345.     {
  1346.       Fvertical_motion (make_number (- ht / 2));
  1347.       XFASTINT (tem) = point;
  1348.       Fset_marker (w->start, tem, w->buffer);
  1349.       w->force_start = Qt;
  1350.     }
  1351.  
  1352.   SET_PT (marker_position (w->start));
  1353.   lose = n < 0 && point == BEGV;
  1354.   Fvertical_motion (make_number (n));
  1355.   pos = point;
  1356.   bolp = Fbolp ();
  1357.   SET_PT (opoint);
  1358.  
  1359.   if (lose)
  1360.     {
  1361.       if (noerror)
  1362.     return Qbeginning_of_buffer;
  1363.       Fsignal (Qbeginning_of_buffer, Qnil);
  1364.     }
  1365.  
  1366.   if (pos < ZV)
  1367.     {
  1368.       set_marker_restricted (w->start, make_number (pos), w->buffer);
  1369.       w->start_at_line_beg = bolp;
  1370.       w->update_mode_line = Qt;
  1371.       XFASTINT (w->last_modified) = 0;
  1372.       if (pos > opoint)
  1373.     SET_PT (pos);
  1374.       if (n < 0)
  1375.     {
  1376.       SET_PT (pos);
  1377.       tem = Fvertical_motion (make_number (ht));
  1378.       if (point > opoint || XFASTINT (tem) < ht)
  1379.         SET_PT (opoint);
  1380.       else
  1381.         Fvertical_motion (make_number (-1));
  1382.     }
  1383.       return Qnil;
  1384.     }
  1385.   else
  1386.     {
  1387.       if (!noerror)
  1388.     Fsignal (Qend_of_buffer, Qnil);
  1389.       return Qend_of_buffer;
  1390.     }
  1391. }
  1392.  
  1393. scroll_command (n, direction)
  1394.      register Lisp_Object n;
  1395.      int direction;
  1396. {
  1397.   register int defalt
  1398.     = direction * (window_height (selected_window) - 1
  1399.            - next_screen_context_lines);
  1400.  
  1401.   if (NULL (n))
  1402.     window_scroll (selected_window, defalt, 0);
  1403.   else if (EQ (n, Qminus))
  1404.     window_scroll (selected_window, - defalt, 0);
  1405.   else
  1406.     {
  1407.       n = Fprefix_numeric_value (n);
  1408.       window_scroll (selected_window, XINT (n) * direction, 0);
  1409.     }
  1410. }
  1411.  
  1412. DEFUN ("scroll-up", Fscroll_up, Sscroll_up, 0, 1, "P",
  1413.   "Scroll text of current window upward ARG lines; or near full screen if no ARG.\n\
  1414. When calling from a program, supply a number as argument or nil.")
  1415.   (n)
  1416.      Lisp_Object n;
  1417. {
  1418.   scroll_command (n, 1);
  1419.   return Qnil;
  1420. }
  1421.  
  1422. DEFUN ("scroll-down", Fscroll_down, Sscroll_down, 0, 1, "P",
  1423.   "Scroll text of current window downward ARG lines; or near full screen if no ARG.\n\
  1424. When calling from a program, supply a number as argument or nil.")
  1425.   (n)
  1426.      Lisp_Object n;
  1427. {
  1428.   scroll_command (n, -1);
  1429.   return Qnil;
  1430. }
  1431.  
  1432. DEFUN ("scroll-left", Fscroll_left, Sscroll_left, 1, 1, "P",
  1433.   "Scroll selected window display ARG columns left.\n\
  1434. Default for ARG is window width minus 2.")
  1435.   (arg)
  1436.      register Lisp_Object arg;
  1437. {
  1438.   if (NULL (arg))
  1439.     XFASTINT (arg) = XFASTINT (XWINDOW (selected_window)->width) - 2;
  1440.   else
  1441.     arg = Fprefix_numeric_value (arg);
  1442.  
  1443.   return Fset_window_hscroll (selected_window,
  1444.                   make_number (XINT (XWINDOW (selected_window)->hscroll)
  1445.                        + XINT (arg)));
  1446. }
  1447.  
  1448. DEFUN ("scroll-right", Fscroll_right, Sscroll_right, 1, 1, "P",
  1449.   "Scroll selected window display ARG columns right.\n\
  1450. Default for ARG is window width minus 2.")
  1451.   (arg)
  1452.      register Lisp_Object arg;
  1453. {
  1454.   if (NULL (arg))
  1455.     XFASTINT (arg) = XFASTINT (XWINDOW (selected_window)->width) - 2;
  1456.   else
  1457.     arg = Fprefix_numeric_value (arg);
  1458.  
  1459.   return Fset_window_hscroll (selected_window,
  1460.                   make_number (XINT (XWINDOW (selected_window)->hscroll)
  1461.                        - XINT (arg)));
  1462. }
  1463.  
  1464. DEFUN ("scroll-other-window", Fscroll_other_window, Sscroll_other_window, 0, 1, "P",
  1465.   "Scroll text of next window upward ARG lines; or near full screen if no ARG.\n\
  1466. The next window is the one below the current one; or the one at the top\n\
  1467. if the current one is at the bottom.\n\
  1468. When calling from a program, supply a number as argument or nil.")
  1469.   (n)
  1470.      register Lisp_Object n;
  1471. {
  1472.   register Lisp_Object window;
  1473.   struct buffer *old = current_buffer;
  1474.   register int ht;
  1475.   register int opoint = point;
  1476.   register struct window *w;
  1477.   Lisp_Object result;
  1478.  
  1479.   if (EQ (selected_window, minibuf_window)
  1480.       && !NULL (Vminibuf_scroll_window))
  1481.     window = Vminibuf_scroll_window;
  1482.   else
  1483.     window = Fnext_window (selected_window, Qnil);
  1484.   CHECK_WINDOW (window, 0);
  1485.   ht = window_height (window) - 1;
  1486.  
  1487.   if (EQ (window, selected_window))
  1488.     error ("There is no other window");
  1489.  
  1490.   w = XWINDOW (window);
  1491.   Fset_buffer (w->buffer);
  1492.   SET_PT (marker_position (w->pointm));
  1493.  
  1494.   if (NULL (n))
  1495.     result = window_scroll (window, ht - next_screen_context_lines, 1);
  1496.   else if (EQ (n, Qminus))
  1497.     result = window_scroll (window, next_screen_context_lines - ht, 1);
  1498.   else
  1499.     {
  1500.       if (XTYPE (n) == Lisp_Cons)
  1501.     n = Fcar (n);
  1502.       CHECK_NUMBER (n, 0);
  1503.       result = window_scroll (window, XINT (n), 1);
  1504.     }
  1505.  
  1506.   Fset_marker (w->pointm, make_number (point), Qnil);
  1507.   set_buffer_internal (old);
  1508.   SET_PT (opoint);
  1509.   if (!EQ (result, Qnil))
  1510.     Fsignal (result, Qnil);    
  1511.   return Qnil;
  1512. }
  1513.  
  1514. DEFUN ("recenter", Frecenter, Srecenter, 0, 1, "P",
  1515.   "Center point in window and redisplay screen.  With ARG, put point on line ARG.\n\
  1516. The desired position of point is always relative to the current window.\n\
  1517. Just C-u as prefix means put point in the center of the screen.\n\
  1518. No arg (i.e., it is nil) erases the entire screen and then\n\
  1519. redraws with point in the center.")
  1520.   (n)
  1521.      register Lisp_Object n;
  1522. {
  1523.   register int ht = window_height (selected_window)
  1524.     - !EQ (selected_window, minibuf_window);
  1525.   register struct window *w = XWINDOW (selected_window);
  1526.   register int opoint = point;
  1527.  
  1528.   if (NULL (n))
  1529.     {
  1530.       extern int screen_garbaged;
  1531.       screen_garbaged++;
  1532.       XFASTINT (n) = ht / 2;
  1533.     }
  1534.   else if (XTYPE (n) == Lisp_Cons) /* Just C-u. */
  1535.     {
  1536.       XFASTINT (n) = ht / 2;
  1537.     }
  1538.   else
  1539.     {
  1540.       n = Fprefix_numeric_value (n);
  1541.       CHECK_NUMBER (n, 0);
  1542.     }
  1543.  
  1544.   if (XINT (n) < 0)
  1545.     XSETINT (n, XINT (n) + ht);
  1546.  
  1547.   XSETINT (n, - XINT (n));
  1548.  
  1549.   Fvertical_motion (n);
  1550.   Fset_marker (w->start, make_number (point), w->buffer);
  1551.   w->start_at_line_beg = Fbolp ();
  1552.  
  1553.   SET_PT (opoint);
  1554.   w->force_start = Qt;
  1555.  
  1556.   return Qnil;
  1557. }
  1558.  
  1559. DEFUN ("move-to-window-line", Fmove_to_window_line, Smove_to_window_line,
  1560.   1, 1, "P",
  1561.   "Position point relative to window.\n\
  1562. With no argument, position at text at center of window.\n\
  1563. An argument specifies screen line; zero means top of window,\n\
  1564. negative means relative to bottom of window.")
  1565.   (arg)
  1566.      register Lisp_Object arg;
  1567. {
  1568.   register struct window *w = XWINDOW (selected_window);
  1569.   register int height = XFASTINT (w->height);
  1570.   register int start;
  1571.  
  1572.   if (!EQ (selected_window, minibuf_window)) height--;
  1573.  
  1574.   if (NULL (arg))
  1575.     XFASTINT (arg) = height / 2;
  1576.   else
  1577.     {
  1578.       arg = Fprefix_numeric_value (arg);
  1579.       if (XINT (arg) < 0)
  1580.     XSETINT (arg, XINT (arg) + height);
  1581.     }
  1582.  
  1583.   start = marker_position (w->start);
  1584.   if (start < BEGV || start > ZV)
  1585.     {
  1586.       Fvertical_motion (make_number (- height / 2));
  1587.       Fset_marker (w->start, make_number (point), w->buffer);
  1588.       w->start_at_line_beg = Fbolp ();
  1589.       w->force_start = Qt;
  1590.     }
  1591.   else
  1592.     SET_PT (start);
  1593.  
  1594.   return Fvertical_motion (arg);
  1595. }
  1596.  
  1597. struct save_window_data
  1598.   {
  1599.     int size_from_Lisp_Vector_struct;
  1600.     struct Lisp_Vector *next_from_Lisp_Vector_struct;
  1601.     Lisp_Object screen_width, screen_height;
  1602.     Lisp_Object current_window;
  1603.     Lisp_Object current_buffer;
  1604.     Lisp_Object minibuf_scroll_window;
  1605.     /* A vector, interpreted as a struct saved_window */
  1606.     Lisp_Object saved_windows;
  1607.   };
  1608. #define SAVE_WINDOW_DATA_SIZE 6 /* Arg to Fmake_vector */
  1609.  
  1610. /* This is saved as a Lisp_Vector */
  1611. struct saved_window
  1612.   {
  1613.     /* these first two must agree with struct Lisp_Vector in lisp.h */
  1614.     int size_from_Lisp_Vector_struct;
  1615.     struct Lisp_Vector *next_from_Lisp_Vector_struct;
  1616.  
  1617.     Lisp_Object window;
  1618.     Lisp_Object buffer, start, pointm, mark;
  1619.     Lisp_Object left, top, width, height, hscroll;
  1620.     Lisp_Object parent, prev;
  1621.     Lisp_Object start_at_line_beg;
  1622.   };
  1623. #define SAVED_WINDOW_VECTOR_SIZE 13 /* Arg to Fmake_vector */
  1624.  
  1625. #define SAVED_WINDOW_N(swv,n) \
  1626.   ((struct saved_window *) (XVECTOR ((swv)->contents[(n)])))
  1627.  
  1628. DEFUN ("set-window-configuration",
  1629.        Fset_window_configuration, Sset_window_configuration,
  1630.        1, 1, 0,
  1631.        "Restore the configuration of Emacs' windows and buffers to\n\
  1632. the state specified by CONFIGURATION.  CONFIGURATION must be a value\n\
  1633. retrned by  current-window-configuration  -- see the documentation of that\n\
  1634. function for more information.")
  1635.      (arg)
  1636.      Lisp_Object arg;
  1637. {
  1638.   register struct window *w;
  1639.   register struct save_window_data *data;
  1640.   struct Lisp_Vector *saved_windows;
  1641.   register struct saved_window *p;
  1642.   register Lisp_Object tem;
  1643.   Lisp_Object new_current_buffer;
  1644.   int k;
  1645.  
  1646.   /* Save screen height here so we can go back to it at the end.  */
  1647.   int previous_screen_height = screen_height;
  1648.   int previous_screen_width = screen_width;
  1649.   int screen_size_change = 0;
  1650.  
  1651.   while (XTYPE (arg) != Lisp_Window_Configuration)
  1652.     {
  1653.       /* the function window-configuration-p isn't actually defined
  1654.      at present --- is there a need for it? */
  1655.       arg = wrong_type_argument (intern ("window-configuration-p"), arg);
  1656.     }
  1657.  
  1658.   data = (struct save_window_data *) XVECTOR (arg);
  1659.   saved_windows = XVECTOR (data->saved_windows);
  1660.  
  1661.   /* Set the screen height to the value it had at save time.  */
  1662.   if (XFASTINT (data->screen_height) != screen_height
  1663.       || XFASTINT (data->screen_width) != screen_width)
  1664.     {
  1665.       change_screen_size (data->screen_height, data->screen_width, 0, 0, 0);
  1666.       screen_size_change = 1;
  1667.     }
  1668.  
  1669.   windows_or_buffers_changed++;
  1670.   new_current_buffer = data->current_buffer;
  1671.   if (NULL (XBUFFER (new_current_buffer)->name))
  1672.     new_current_buffer = Qnil;
  1673.  
  1674.   for (k = 0; k < saved_windows->size; k++)
  1675.     {
  1676.       p = SAVED_WINDOW_N (saved_windows, k);
  1677.       w = XWINDOW (p->window);
  1678.       w->next = Qnil;
  1679.  
  1680.       if (!NULL (p->parent))
  1681.     w->parent = SAVED_WINDOW_N (saved_windows, XFASTINT (p->parent))->window;
  1682.       else
  1683.     w->parent = Qnil;
  1684.  
  1685.       if (!NULL (p->prev))
  1686.     {
  1687.       w->prev = SAVED_WINDOW_N (saved_windows, XFASTINT (p->prev))->window;
  1688.       XWINDOW (w->prev)->next = p->window;
  1689.     }
  1690.       else
  1691.     {
  1692.       w->prev = Qnil;
  1693.       if (!NULL (w->parent))
  1694.         {
  1695.           if (EQ (p->width, XWINDOW (w->parent)->width))
  1696.         {
  1697.           XWINDOW (w->parent)->vchild = p->window;
  1698.           XWINDOW (w->parent)->hchild = Qnil;
  1699.         }
  1700.           else
  1701.         {
  1702.           XWINDOW (w->parent)->hchild = p->window;
  1703.           XWINDOW (w->parent)->vchild = Qnil;
  1704.         }
  1705.         }
  1706.     }
  1707.       w->left = p->left;
  1708.       w->top = p->top;
  1709.       w->width = p->width;
  1710.       w->height = p->height;
  1711.       w->hscroll = p->hscroll;
  1712.       XFASTINT (w->last_modified) = 0;
  1713.  
  1714.       /* Reinstall the saved buffer and pointers into it.  */
  1715.       if (NULL (p->buffer))
  1716.     w->buffer = p->buffer;
  1717.       else
  1718.     {
  1719.       if (!NULL (XBUFFER (p->buffer)->name))
  1720.         /* If saved buffer is alive, install it.  */
  1721.         {
  1722.           w->buffer = p->buffer;
  1723.           w->start_at_line_beg = p->start_at_line_beg;
  1724.           set_marker_restricted (w->start,
  1725.                      Fmarker_position (p->start), w->buffer);
  1726.           set_marker_restricted (w->pointm,
  1727.                      Fmarker_position (p->pointm), w->buffer);
  1728.           Fset_marker (XBUFFER (w->buffer)->mark,
  1729.                Fmarker_position (p->mark), w->buffer);
  1730.  
  1731.           if (!EQ (p->buffer, new_current_buffer) &&
  1732.           XBUFFER (p->buffer) == current_buffer)
  1733.         Fgoto_char (w->pointm);
  1734.         }
  1735.       else if (NULL (XBUFFER (w->buffer)->name))
  1736.         /* Else if window's old buffer is dead too, get a live one.  */
  1737.         {
  1738.           w->buffer = Fcdr (Fcar (Vbuffer_alist));
  1739.           /* Set window markers at start of buffer.
  1740.          Rely on set_marker_restricted to put them
  1741.          within the restriction.  */
  1742.           set_marker_restricted (w->start, make_number (0), w->buffer);
  1743.           set_marker_restricted (w->pointm, make_number (0), w->buffer);
  1744.           w->start_at_line_beg = Qt;
  1745.         }
  1746.       else
  1747.         /* Keeping window's old buffer; make sure the markers are real.  */
  1748.         /* Else if window's old buffer is dead too, get a live one.  */
  1749.         {
  1750.           /* Set window markers at start of buffer.
  1751.          Rely on set_marker_restricted to put them within the restriction.  */
  1752.           if (XMARKER (w->start)->buffer == 0)
  1753.         set_marker_restricted (w->start, make_number (0), w->buffer);
  1754.           if (XMARKER (w->pointm)->buffer == 0)
  1755.         set_marker_restricted (w->pointm,
  1756.                        make_number (BUF_PT (XBUFFER (w->buffer))),
  1757.                        w->buffer);
  1758.           w->start_at_line_beg = Qt;
  1759.         }
  1760.     }
  1761.     }
  1762.  
  1763.   /* Set the screen height to the value it had before this function.  */
  1764.   if (screen_size_change)
  1765.     change_screen_size (previous_screen_height, previous_screen_width, 0, 0, 0);
  1766.  
  1767.   Fselect_window (data->current_window);
  1768.   if (!NULL (new_current_buffer))
  1769.     Fset_buffer (new_current_buffer);
  1770.   else
  1771.     Fset_buffer (XWINDOW (selected_window)->buffer);
  1772.   Vminibuf_scroll_window = data->minibuf_scroll_window;
  1773.   return (Qnil);
  1774. }
  1775.  
  1776.  
  1777. static int
  1778. count_windows (window)
  1779.      register struct window *window;
  1780. {
  1781.   register int count = 1;
  1782.   if (!NULL (window->next))
  1783.     count += count_windows (XWINDOW (window->next));
  1784.   if (!NULL (window->vchild))
  1785.     count += count_windows (XWINDOW (window->vchild));
  1786.   if (!NULL (window->hchild))
  1787.     count += count_windows (XWINDOW (window->hchild));
  1788.   return count;
  1789. }
  1790.  
  1791. DEFUN ("current-window-configuration",
  1792.     Fcurrent_window_configuration, Scurrent_window_configuration, 0, 0, 0,
  1793.        "Return an object representing Emacs' current window configuration,\n\
  1794. namely the number of windows, their sizes and current buffers, and for\n\
  1795. each displayed buffer, where display starts, and the positions of\n\
  1796. point and mark.  An exception is made for point in (current-buffer) --\n\
  1797. its value is -not- saved.")
  1798.   ()
  1799. {
  1800.   register Lisp_Object tem;
  1801.   register int n_windows;
  1802.   register struct save_window_data *data;
  1803.   register int i;
  1804.  
  1805.   n_windows = count_windows (XWINDOW (XWINDOW (minibuf_window)->prev));
  1806.   data = (struct save_window_data *)
  1807.            XVECTOR (Fmake_vector (make_number (SAVE_WINDOW_DATA_SIZE),
  1808.                   Qnil));
  1809.   XFASTINT (data->screen_width) = screen_width;
  1810.   XFASTINT (data->screen_height) = screen_height;
  1811.   data->current_window = selected_window;
  1812.   XSET (data->current_buffer, Lisp_Buffer, current_buffer);
  1813.   data->minibuf_scroll_window = Vminibuf_scroll_window;
  1814.   tem = Fmake_vector (make_number (n_windows), Qnil);
  1815.   data->saved_windows = tem;
  1816.   for (i = 0; i < n_windows; i++)
  1817.     XVECTOR (tem)->contents[i]
  1818.       = Fmake_vector (make_number (SAVED_WINDOW_VECTOR_SIZE), Qnil);
  1819.   save_window_save (XWINDOW (minibuf_window)->prev,
  1820.             XVECTOR (tem),
  1821.             0, n_windows);
  1822.   XSET (tem, Lisp_Window_Configuration, data);
  1823.   return (tem);
  1824. }
  1825.  
  1826. static int
  1827. save_window_save (window, vector, i, maxwindow)
  1828.      Lisp_Object window;
  1829.      struct Lisp_Vector *vector;
  1830.      int i;
  1831.      int maxwindow;
  1832. {
  1833.   register struct saved_window *p;
  1834.   register struct window *w;
  1835.   register Lisp_Object tem;
  1836.  
  1837.   for (;!NULL (window); window = w->next)
  1838.     {
  1839.       /* If you get a crash here, you may be seeing a very weird bug.
  1840.      When it happened to me, it seems that count_windows returned
  1841.      a value that was too small--only two, when there were two
  1842.      visible windows, a parent, and the minibuffer (inactive).
  1843.      If this starts happening for you, please run under a debugger
  1844.      with a breakpoint at the abort, so that you can at least try calling
  1845.      count_windows again to see if it will lose again.
  1846.      If it does, you can find the bug.  */
  1847.       if (i == maxwindow)
  1848.     abort ();
  1849.  
  1850.       p = SAVED_WINDOW_N (vector, i);
  1851.       w = XWINDOW (window);
  1852.  
  1853.       XFASTINT (w->temslot) = i++;
  1854.  
  1855.       p->window = window;
  1856.       p->buffer = w->buffer;
  1857.       p->left = w->left;
  1858.       p->top = w->top;
  1859.       p->width = w->width;
  1860.       p->height = w->height;
  1861.       p->hscroll = w->hscroll;
  1862.       if (!NULL (w->buffer))
  1863.     {
  1864.       /* Save w's value of point in the window configuration.
  1865.          If w is the selected window, then get the value of point
  1866.          from the buffer; pointm is garbage in the selected window.  */
  1867.       if (EQ (window, selected_window))
  1868.         {
  1869.           p->pointm = Fmake_marker ();
  1870.           Fset_marker (p->pointm, BUF_PT (XBUFFER (w->buffer)),
  1871.                w->buffer);
  1872.         }
  1873.       else
  1874.         p->pointm = Fcopy_marker (w->pointm);
  1875.  
  1876.       p->start = Fcopy_marker (w->start);
  1877.       p->start_at_line_beg = w->start_at_line_beg;
  1878.  
  1879.       tem = XBUFFER (w->buffer)->mark;
  1880.       p->mark = Fcopy_marker (tem);
  1881.     }
  1882.       else
  1883.     {
  1884.       p->pointm = Qnil;
  1885.       p->start = Qnil;
  1886.       p->mark = Qnil;
  1887.       p->start_at_line_beg = Qnil;
  1888.     }
  1889.  
  1890.       if (NULL (w->parent))
  1891.     p->parent = Qnil;
  1892.       else
  1893.     p->parent = XWINDOW (w->parent)->temslot;
  1894.  
  1895.       if (NULL (w->prev))
  1896.     p->prev = Qnil;
  1897.       else
  1898.     p->prev = XWINDOW (w->prev)->temslot;
  1899.  
  1900.       if (!NULL (w->vchild))
  1901.     i = save_window_save (w->vchild, vector, i, maxwindow);
  1902.       if (!NULL (w->hchild))
  1903.     i = save_window_save (w->hchild, vector, i, maxwindow);
  1904.     }
  1905.  
  1906.   return i;
  1907. }
  1908.  
  1909. DEFUN ("save-window-excursion", Fsave_window_excursion, Ssave_window_excursion,
  1910.   0, UNEVALLED, 0,
  1911.   "Execute body, preserving window sizes and contents.\n\
  1912. Restores which buffer appears in which window, where display starts,\n\
  1913. as well as the current buffer.\n\
  1914. Does not restore the value of point in current buffer.")
  1915.   (args)
  1916.      Lisp_Object args;
  1917. {
  1918.   register Lisp_Object val;
  1919.   register int count = specpdl_ptr - specpdl;
  1920.  
  1921.   record_unwind_protect (Fset_window_configuration,
  1922.              Fcurrent_window_configuration ());
  1923.   val = Fprogn (args);
  1924.   unbind_to (count);
  1925.   return val;
  1926. }
  1927.  
  1928. init_window_once ()
  1929. {
  1930.   extern Lisp_Object get_minibuffer ();
  1931.   register Lisp_Object root_window;
  1932.  
  1933.   root_window = make_window ();
  1934.   minibuf_window = make_window ();
  1935.  
  1936.   XWINDOW (root_window)->next = minibuf_window;
  1937.   XWINDOW (minibuf_window)->prev = root_window;
  1938.  
  1939.   /* These values 9 and 10 are arbitrary,
  1940.      just so that there is "something there."
  1941.      Correct values are put in in init_xdisp */
  1942.  
  1943.   XFASTINT (XWINDOW (root_window)->width) = 10;
  1944.   XFASTINT (XWINDOW (minibuf_window)->width) = 10;
  1945.  
  1946.   XFASTINT (XWINDOW (root_window)->height) = 9;
  1947.   XFASTINT (XWINDOW (minibuf_window)->top) = 9;
  1948.   XFASTINT (XWINDOW (minibuf_window)->height) = 1;
  1949.  
  1950.   Fset_window_buffer (root_window, Fcurrent_buffer ());
  1951.   Fset_window_buffer (minibuf_window, get_minibuffer (0));
  1952.  
  1953.   selected_window = root_window;
  1954.   /* Make sure this window seems more recently used than
  1955.      a newly-created, never-selected window.  */
  1956.   XFASTINT (XWINDOW (selected_window)->use_time) = ++window_select_count;
  1957. }
  1958.  
  1959. syms_of_window ()
  1960. {
  1961.   Qwindowp = intern ("windowp");
  1962.   staticpro (&Qwindowp);
  1963.  
  1964.   /* Make sure all windows get marked */
  1965.   staticpro (&minibuf_window);
  1966.  
  1967.   DEFVAR_INT ("minibuffer-prompt-width", &minibuf_prompt_width,
  1968.     "Width of the prompt appearing at the start of the minibuffer window.\n\
  1969. The value is meaningless when the minibuffer is not visible.");
  1970.  
  1971.   DEFVAR_LISP ("temp-buffer-show-hook", &Vtemp_buffer_show_hook,
  1972.     "Non-nil means call as function to display a help buffer.\n\
  1973. Used by with-output-to-temp-buffer.");
  1974.   Vtemp_buffer_show_hook = Qnil;
  1975.  
  1976.   DEFVAR_LISP ("minibuffer-scroll-window", &Vminibuf_scroll_window,
  1977.     "Non-nil means it is the window that C-M-v in minibuffer should scroll.");
  1978.   Vminibuf_scroll_window = Qnil;
  1979.  
  1980.   DEFVAR_BOOL ("pop-up-windows", &pop_up_windows,
  1981.     "*Non-nil means display-buffer should make new windows.");
  1982.   pop_up_windows = 1;
  1983.  
  1984.   DEFVAR_INT ("next-screen-context-lines", &next_screen_context_lines,
  1985.     "*Number of lines of continuity when scrolling by screenfuls.");
  1986.   next_screen_context_lines = 2;
  1987.  
  1988.   DEFVAR_INT ("split-height-threshold", &split_height_threshold,
  1989.     "*display-buffer would prefer to split the largest window if this large.\n\
  1990. If there is only one window, it is split regardless of this value.");
  1991.   split_height_threshold = 500;
  1992.  
  1993.   DEFVAR_INT ("window-min-height", &window_min_height,
  1994.     "*Delete any window less than this tall (including its mode line).");
  1995.   window_min_height = 4;
  1996.  
  1997.   DEFVAR_INT ("window-min-width", &window_min_width,
  1998.     "*Delete any window less than this wide.");
  1999.   window_min_width = 10;
  2000.  
  2001.   defsubr (&Sselected_window);
  2002.   defsubr (&Sminibuffer_window);
  2003.   defsubr (&Swindowp);
  2004.   defsubr (&Spos_visible_in_window_p);
  2005.   defsubr (&Swindow_buffer);
  2006.   defsubr (&Swindow_height);
  2007.   defsubr (&Swindow_width);
  2008.   defsubr (&Swindow_hscroll);
  2009.   defsubr (&Sset_window_hscroll);
  2010.   defsubr (&Swindow_edges);
  2011.   defsubr (&Swindow_point);
  2012.   defsubr (&Swindow_start);
  2013.   defsubr (&Sset_window_point);
  2014.   defsubr (&Sset_window_start);
  2015.   defsubr (&Snext_window);
  2016.   defsubr (&Sprevious_window);
  2017.   defsubr (&Sother_window);
  2018.   defsubr (&Sget_lru_window);
  2019.   defsubr (&Sget_largest_window);
  2020.   defsubr (&Sget_buffer_window);
  2021.   defsubr (&Sdelete_other_windows);
  2022.   defsubr (&Sdelete_windows_on);
  2023.   defsubr (&Sreplace_buffer_in_windows);
  2024.   defsubr (&Sdelete_window);
  2025.   defsubr (&Sset_window_buffer);
  2026.   defsubr (&Sselect_window);
  2027.   defsubr (&Sdisplay_buffer);
  2028.   defsubr (&Ssplit_window);
  2029.   defsubr (&Senlarge_window);
  2030.   defsubr (&Sshrink_window);
  2031.   defsubr (&Sscroll_up);
  2032.   defsubr (&Sscroll_down);
  2033.   defsubr (&Sscroll_left);
  2034.   defsubr (&Sscroll_right);
  2035.   defsubr (&Sscroll_other_window);
  2036.   defsubr (&Srecenter);
  2037.   defsubr (&Smove_to_window_line);
  2038.   defsubr (&Sset_window_configuration);
  2039.   defsubr (&Scurrent_window_configuration);
  2040.   defsubr (&Ssave_window_excursion);
  2041. }
  2042.  
  2043. keys_of_window ()
  2044. {
  2045.   ndefkey (Vctl_x_map, '1', "delete-other-windows");
  2046.   ndefkey (Vctl_x_map, '2', "split-window");
  2047.   ndefkey (Vctl_x_map, '0', "delete-window");
  2048.   ndefkey (Vctl_x_map, 'o', "other-window");
  2049.   ndefkey (Vctl_x_map, '^', "enlarge-window");
  2050.   ndefkey (Vctl_x_map, '<', "scroll-left");
  2051.   ndefkey (Vctl_x_map, '>', "scroll-right");
  2052.  
  2053.   ndefkey (Vglobal_map, Ctl ('V'), "scroll-up");
  2054.   ndefkey (Vesc_map, Ctl ('V'), "scroll-other-window");
  2055.   ndefkey (Vesc_map, 'v', "scroll-down");
  2056.  
  2057.   ndefkey (Vglobal_map, Ctl('L'), "recenter");
  2058.   ndefkey (Vesc_map, 'r', "move-to-window-line");
  2059. }
  2060.